All files / web/src/app/api/audio/songs/[songId]/alignment route.ts

0% Statements 0/47
0% Branches 0/1
0% Functions 0/1
0% Lines 0/47

Press n or j to go to the next uncovered block, b, p or k for the previous block.

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48                                                                                               
/**
 * API route for serving session song word-alignment JSON.
 *
 * GET /api/audio/songs/[songId]/alignment
 *
 * Serves the ElevenLabs word-timestamp JSON sidecar written next to the MP3
 * at data/audio/songs/{songId}.json. Returns 404 when no sidecar exists
 * (e.g. legacy songs generated before the timestamps feature shipped).
 */

import { readFile, stat } from 'fs/promises'
import { NextResponse } from 'next/server'
import { join } from 'path'
import { withAuth } from '@/lib/auth/withAuth'

const SONGS_DIR = join(process.cwd(), 'data', 'audio', 'songs')

export const GET = withAuth(async (_request, { params }) => {
  try {
    const { songId } = (await params) as { songId: string }

    if (!songId || songId.includes('/') || songId.includes('..')) {
      return NextResponse.json({ error: 'Invalid parameters' }, { status: 400 })
    }

    const filePath = join(SONGS_DIR, `${songId}.json`)

    try {
      await stat(filePath)
    } catch {
      return new NextResponse(null, { status: 404 })
    }

    const fileBuffer = await readFile(filePath)

    return new NextResponse(new Uint8Array(fileBuffer), {
      headers: {
        'Content-Type': 'application/json',
        'Content-Length': fileBuffer.byteLength.toString(),
        'Cache-Control': 'public, max-age=31536000, immutable',
      },
    })
  } catch (error) {
    console.error('Error serving song alignment:', error)
    return new NextResponse(null, { status: 500 })
  }
})